home *** CD-ROM | disk | FTP | other *** search
/ Micromanía 93 / CDMM_93_2.ISO / Project Nomads / nomads_demo_eng.exe / MSGCAT.TCL < prev    next >
Encoding:
Text File  |  2000-12-15  |  8.0 KB  |  303 lines

  1. # msgcat.tcl --
  2. #
  3. #    This file defines various procedures which implement a
  4. #    message catalog facility for Tcl programs.  It should be
  5. #    loaded with the command "package require msgcat".
  6. #
  7. # Copyright (c) 1998-2000 by Ajuba Solutions.
  8. # Copyright (c) 1998 by Mark Harrison.
  9. #
  10. # See the file "license.terms" for information on usage and redistribution
  11. # of this file, and for a DISCLAIMER OF ALL WARRANTIES.
  12. # RCS: @(#) $Id: msgcat.tcl,v 1.1 2000/12/15 20:10:58 floh Exp $
  13.  
  14. package provide msgcat 1.2
  15.  
  16. namespace eval msgcat {
  17.     namespace export mc mcset mcmset mclocale mcpreferences mcunknown mcmax
  18.  
  19.     # Records the current locale as passed to mclocale
  20.     variable locale ""
  21.  
  22.     # Records the list of locales to search
  23.     variable loclist {}
  24.  
  25.     # Records the mapping between source strings and translated strings.  The
  26.     # array key is of the form "<locale>,<namespace>,<src>" and the value is
  27.     # the translated string.
  28.     array set msgs {}
  29. }
  30.  
  31. # msgcat::mc --
  32. #
  33. #    Find the translation for the given string based on the current
  34. #    locale setting. Check the local namespace first, then look in each
  35. #    parent namespace until the source is found.  If additional args are
  36. #    specified, use the format command to work them into the traslated
  37. #    string.
  38. #
  39. # Arguments:
  40. #    src    The string to translate.
  41. #    args    Args to pass to the format command
  42. #
  43. # Results:
  44. #    Returns the translatd string.  Propagates errors thrown by the 
  45. #    format command.
  46.  
  47. proc msgcat::mc {src args} {
  48.     # Check for the src in each namespace starting from the local and
  49.     # ending in the global.
  50.  
  51.     set ns [uplevel {namespace current}]
  52.     
  53.     while {$ns != ""} {
  54.     foreach loc $::msgcat::loclist {
  55.         if {[info exists ::msgcat::msgs($loc,$ns,$src)]} {
  56.         if {[llength $args] == 0} {
  57.             return $::msgcat::msgs($loc,$ns,$src)
  58.         } else {
  59.             return [eval \
  60.                 [list format $::msgcat::msgs($loc,$ns,$src)] \
  61.                 $args]
  62.         }
  63.         }
  64.     }
  65.     set ns [namespace parent $ns]
  66.     }
  67.     # we have not found the translation
  68.     return [uplevel 1 [list [namespace origin mcunknown] \
  69.         $::msgcat::locale $src] $args]
  70. }
  71.  
  72. # msgcat::mclocale --
  73. #
  74. #    Query or set the current locale.
  75. #
  76. # Arguments:
  77. #    newLocale    (Optional) The new locale string. Locale strings
  78. #            should be composed of one or more sublocale parts
  79. #            separated by underscores (e.g. en_US).
  80. #
  81. # Results:
  82. #    Returns the current locale.
  83.  
  84. proc msgcat::mclocale {args} {
  85.     set len [llength $args]
  86.  
  87.     if {$len > 1} {
  88.     error {wrong # args: should be "mclocale ?newLocale?"}
  89.     }
  90.  
  91.     set args [string tolower $args]
  92.     if {$len == 1} {
  93.     set ::msgcat::locale $args
  94.     set ::msgcat::loclist {}
  95.     set word ""
  96.     foreach part [split $args _] {
  97.         set word [string trimleft "${word}_${part}" _]
  98.         set ::msgcat::loclist [linsert $::msgcat::loclist 0 $word]
  99.     }
  100.     }
  101.     return $::msgcat::locale
  102. }
  103.  
  104. # msgcat::mcpreferences --
  105. #
  106. #    Fetch the list of locales used to look up strings, ordered from
  107. #    most preferred to least preferred.
  108. #
  109. # Arguments:
  110. #    None.
  111. #
  112. # Results:
  113. #    Returns an ordered list of the locales preferred by the user.
  114.  
  115. proc msgcat::mcpreferences {} {
  116.     return $::msgcat::loclist
  117. }
  118.  
  119. # msgcat::mcload --
  120. #
  121. #    Attempt to load message catalogs for each locale in the
  122. #    preference list from the specified directory.
  123. #
  124. # Arguments:
  125. #    langdir        The directory to search.
  126. #
  127. # Results:
  128. #    Returns the number of message catalogs that were loaded.
  129.  
  130. proc msgcat::mcload {langdir} {
  131.     set x 0
  132.     foreach p [::msgcat::mcpreferences] {
  133.     set langfile [file join $langdir $p.msg]
  134.     if {[file exists $langfile]} {
  135.         incr x
  136.         set fid [open $langfile "r"]
  137.         fconfigure $fid -encoding utf-8
  138.             uplevel [list eval [read $fid]]
  139.         close $fid
  140.     }
  141.     }
  142.     return $x
  143. }
  144.  
  145. # msgcat::mcset --
  146. #
  147. #    Set the translation for a given string in a specified locale.
  148. #
  149. # Arguments:
  150. #    locale        The locale to use.
  151. #    src        The source string.
  152. #    dest        (Optional) The translated string.  If omitted,
  153. #            the source string is used.
  154. #
  155. # Results:
  156. #    Returns the new locale.
  157.  
  158. proc msgcat::mcset {locale src {dest ""}} {
  159.     if {[string equal $dest ""]} {
  160.     set dest $src
  161.     }
  162.  
  163.     set ns [uplevel {namespace current}]
  164.  
  165.     set ::msgcat::msgs([string tolower $locale],$ns,$src) $dest
  166.     return $dest
  167. }
  168.  
  169. # msgcat::mcmset --
  170. #
  171. #    Set the translation for multiple strings in a specified locale.
  172. #
  173. # Arguments:
  174. #    locale        The locale to use.
  175. #    pairs        One or more src/dest pairs (must be even length)
  176. #
  177. # Results:
  178. #    Returns the number of pairs processed
  179.  
  180. proc msgcat::mcmset {locale pairs } {
  181.  
  182.     set length [llength $pairs]
  183.     if {$length % 2} {
  184.     error {bad translation list: should be "mcmset locale {src dest ...}"}
  185.     }
  186.     
  187.     set locale [string tolower $locale]
  188.     set ns [uplevel {namespace current}]
  189.     
  190.     foreach {src dest} $pairs {
  191.         set ::msgcat::msgs($locale,$ns,$src) $dest
  192.     }
  193.     
  194.     return $length
  195. }
  196.  
  197. # msgcat::mcunknown --
  198. #
  199. #    This routine is called by msgcat::mc if a translation cannot
  200. #    be found for a string.  This routine is intended to be replaced
  201. #    by an application specific routine for error reporting
  202. #    purposes.  The default behavior is to return the source string.  
  203. #    If additional args are specified, the format command will be used
  204. #    to work them into the traslated string.
  205. #
  206. # Arguments:
  207. #    locale        The current locale.
  208. #    src        The string to be translated.
  209. #    args        Args to pass to the format command
  210. #
  211. # Results:
  212. #    Returns the translated value.
  213.  
  214. proc msgcat::mcunknown {locale src args} {
  215.     if {[llength $args]} {
  216.     return [eval [list format $src] $args]
  217.     } else {
  218.     return $src
  219.     }
  220. }
  221.  
  222. # msgcat::mcmax --
  223. #
  224. #    Calculates the maximun length of the translated strings of the given 
  225. #    list.
  226. #
  227. # Arguments:
  228. #    args    strings to translate.
  229. #
  230. # Results:
  231. #    Returns the length of the longest translated string.
  232.  
  233. proc msgcat::mcmax {args} {
  234.     set max 0
  235.     foreach string $args {
  236.         set len [string length [msgcat::mc $string]]
  237.         if {$len>$max} {
  238.             set max $len
  239.         }
  240.     }
  241.     return $max
  242. }
  243.  
  244. # Initialize the default locale
  245.  
  246. namespace eval msgcat {
  247.     # set default locale, try to get from environment
  248.     if {[info exists ::env(LANG)]} {
  249.         mclocale $::env(LANG)
  250.     } else {
  251.         if { $tcl_platform(platform) == "windows" } {
  252.             # try to set locale depending on registry settings
  253.             #
  254.             set key {HKEY_CURRENT_USER\Control Panel\International}
  255.             if {[catch {package require registry}] || \
  256.             [catch {registry get $key "locale"} locale]} {
  257.                 mclocale "C"
  258.             } else {
  259.         
  260.                 #
  261.                 # Clean up registry value for translating LCID value
  262.                 # by using only the last 2 digits, since first
  263.                 # 2 digits appear to be the country...  For example
  264.                 #     0409 - English - United States
  265.                 #     0809 - English - United Kingdom
  266.                 #
  267.                 set locale [string trimleft $locale "0"]
  268.                 set locale [string range $locale end-1 end]
  269.                 set locale [string tolower $locale]
  270.                 switch -- $locale {
  271.             01      { mclocale "ar" }
  272.             02      { mclocale "bg" }
  273.             03      { mclocale "ca" }
  274.             04      { mclocale "zh" }
  275.             05      { mclocale "cs" }
  276.             06      { mclocale "da" }
  277.             07      { mclocale "de" }
  278.             08      { mclocale "el" }
  279.             09      { mclocale "en" }
  280.             0a      { mclocale "es" }
  281.             0b      { mclocale "fi" }
  282.             0c      { mclocale "fr" }
  283.             0d      { mclocale "he" }
  284.             0e      { mclocale "hu" }
  285.             0f      { mclocale "is" }
  286.             10      { mclocale "it" }
  287.             11      { mclocale "ja" }
  288.             12      { mclocale "ko" }
  289.             13      { mclocale "da" }
  290.             14      { mclocale "no" }
  291.             15      { mclocale "pl" }
  292.             16      { mclocale "pt" }
  293.             
  294.             default  { mclocale "C" }
  295.         }
  296.             }
  297.         } else {
  298.             mclocale "C"
  299.         }
  300.     }
  301. }
  302.